<!DOCTYPE html>
<!-- saved from url=(0046)https://habr.com/ru/company/ruvds/blog/328436/ -->
<html lang="ru"><head><meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
  
  <meta name="viewport" content="width=device-width,initial-scale=1.0,viewport-fit=cover">
  <title>Bash-скрипты, часть 11: expect и автоматизация интерактивных утилит / Блог компании RUVDS.com / Хабр</title>
  <style>
    /* cyrillic-ext */
    @font-face {
      font-family: 'Fira Sans';
      font-style: normal;
      font-weight: 500;
      font-display: swap;
      src: url(https://fonts.gstatic.com/s/firasans/v11/va9B4kDNxMZdWfMOD5VnZKveSxf6TF0.woff2) format('woff2');
      unicode-range: U+0460-052F, U+1C80-1C88, U+20B4, U+2DE0-2DFF, U+A640-A69F, U+FE2E-FE2F;
    }

    /* cyrillic */
    @font-face {
      font-family: 'Fira Sans';
      font-style: normal;
      font-weight: 500;
      font-display: swap;
      src: url(https://fonts.gstatic.com/s/firasans/v11/va9B4kDNxMZdWfMOD5VnZKveQhf6TF0.woff2) format('woff2');
      unicode-range: U+0400-045F, U+0490-0491, U+04B0-04B1, U+2116;
    }

    /* latin-ext */
    @font-face {
      font-family: 'Fira Sans';
      font-style: normal;
      font-weight: 500;
      font-display: swap;
      src: url(https://fonts.gstatic.com/s/firasans/v11/va9B4kDNxMZdWfMOD5VnZKveSBf6TF0.woff2) format('woff2');
      unicode-range: U+0100-024F, U+0259, U+1E00-1EFF, U+2020, U+20A0-20AB, U+20AD-20CF, U+2113, U+2C60-2C7F, U+A720-A7FF;
    }

    /* latin */
    @font-face {
      font-family: 'Fira Sans';
      font-style: normal;
      font-weight: 500;
      font-display: swap;
      src: url(https://fonts.gstatic.com/s/firasans/v11/va9B4kDNxMZdWfMOD5VnZKveRhf6.woff2) format('woff2');
      unicode-range: U+0000-00FF, U+0131, U+0152-0153, U+02BB-02BC, U+02C6, U+02DA, U+02DC, U+2000-206F, U+2074, U+20AC, U+2122, U+2191, U+2193, U+2212, U+2215, U+FEFF, U+FFFD;
    }
  </style>
  <link rel="preload" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-vendors.29e3fe9b.css" as="style"><link rel="preload" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-vendors.1ca41abb.js" as="script"><link rel="preload" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/app.b84b16a4.css" as="style"><link rel="preload" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/app.132461d8.js" as="script"><link rel="preload" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/about.428e0545.css" as="style"><link rel="preload" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/about.2d3fd602.js" as="script">
  <link rel="stylesheet" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-vendors.29e3fe9b.css"><link rel="stylesheet" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/app.b84b16a4.css"><link rel="stylesheet" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/about.428e0545.css">
  <script type="text/javascript" async="" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/js"></script><script async="" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/tag.js"></script><script async="" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/analytics.js"></script><script>window.i18nFetch = new Promise((res, rej) => {
          const xhr = new XMLHttpRequest();
          xhr.open('GET', '/js/i18n/ru-compiled.07ab3411a33038d0c6f862dbea0a2c45.json');
          xhr.responseType = 'json';
          xhr.onload = function(e) {
            if (this.status === 200) {
              res({ru: xhr.response});
            } else {
              rej(e);
            }
          };
          xhr.send();
        });</script>
  
  
  <script src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/gpt.js" async=""></script>
  <style>.grecaptcha-badge{visibility: hidden;}</style>
  <meta name="habr-version" content="2.45.0">
  
  <meta data-vue-meta="ssr" property="fb:app_id" content="444736788986613"><meta data-vue-meta="ssr" property="fb:pages" content="472597926099084"><meta data-vue-meta="ssr" name="twitter:card" content="summary_large_image"><meta data-vue-meta="ssr" name="twitter:site" content="@habr_eng"><meta data-vue-meta="ssr" property="og:locale" content="ru_RU" data-vmid="og:locale">
  
  <meta name="apple-mobile-web-app-status-bar-style" content="#303b44">
  <meta name="msapplication-TileColor" content="#629FBC">
  <meta name="apple-mobile-web-app-capable" content="yes">
  <meta name="mobile-web-app-capable" content="yes">
  <link rel="shortcut icon" type="image/png" sizes="16x16" href="https://assets.habr.com/habr-web/img/favicons/favicon-16.png">
  <link rel="shortcut icon" type="image/png" sizes="32x32" href="https://assets.habr.com/habr-web/img/favicons/favicon-32.png">
  <link rel="apple-touch-icon" type="image/png" sizes="76x76" href="https://assets.habr.com/habr-web/img/favicons/apple-touch-icon-76.png">
  <link rel="apple-touch-icon" type="image/png" sizes="120x120" href="https://assets.habr.com/habr-web/img/favicons/apple-touch-icon-120.png">
  <link rel="apple-touch-icon" type="image/png" sizes="152x152" href="https://assets.habr.com/habr-web/img/favicons/apple-touch-icon-152.png">
  <link rel="apple-touch-icon" type="image/png" sizes="180x180" href="https://assets.habr.com/habr-web/img/favicons/apple-touch-icon-180.png">
  <link rel="apple-touch-icon" type="image/png" sizes="256x256" href="https://assets.habr.com/habr-web/img/favicons/apple-touch-icon-256.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 320px) and (device-height: 568px) and (-webkit-device-pixel-ratio: 2) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_1136x640.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 375px) and (device-height: 812px) and (-webkit-device-pixel-ratio: 3) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_2436x1125.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 414px) and (device-height: 896px) and (-webkit-device-pixel-ratio: 2) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_1792x828.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 414px) and (device-height: 896px) and (-webkit-device-pixel-ratio: 2) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_828x1792.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 375px) and (device-height: 667px) and (-webkit-device-pixel-ratio: 2) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_1334x750.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 414px) and (device-height: 896px) and (-webkit-device-pixel-ratio: 3) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_1242x2668.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 414px) and (device-height: 736px) and (-webkit-device-pixel-ratio: 3) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_2208x1242.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 375px) and (device-height: 812px) and (-webkit-device-pixel-ratio: 3) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_1125x2436.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 414px) and (device-height: 736px) and (-webkit-device-pixel-ratio: 3) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_1242x2208.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 1024px) and (device-height: 1366px) and (-webkit-device-pixel-ratio: 2) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_2732x2048.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 414px) and (device-height: 896px) and (-webkit-device-pixel-ratio: 3) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_2688x1242.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 834px) and (device-height: 1112px) and (-webkit-device-pixel-ratio: 2) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_2224x1668.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 375px) and (device-height: 667px) and (-webkit-device-pixel-ratio: 2) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_750x1334.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 1024px) and (device-height: 1366px) and (-webkit-device-pixel-ratio: 2) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_2048x2732.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 834px) and (device-height: 1194px) and (-webkit-device-pixel-ratio: 2) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_2388x1668.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 834px) and (device-height: 1112px) and (-webkit-device-pixel-ratio: 2) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_1668x2224.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 320px) and (device-height: 568px) and (-webkit-device-pixel-ratio: 2) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_640x1136.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 834px) and (device-height: 1194px) and (-webkit-device-pixel-ratio: 2) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_1668x2388.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 768px) and (device-height: 1024px) and (-webkit-device-pixel-ratio: 2) and (orientation: landscape)" href="https://assets.habr.com/habr-web/img/splashes/splash_2048x1536.png">
  <link rel="apple-touch-startup-image" media="screen and (device-width: 768px) and (device-height: 1024px) and (-webkit-device-pixel-ratio: 2) and (orientation: portrait)" href="https://assets.habr.com/habr-web/img/splashes/splash_1536x2048.png">
  <link rel="mask-icon" color="#77a2b6" href="https://assets.habr.com/habr-web/img/favicons/apple-touch-icon-120.svg">
  <link crossorigin="use-credentials" href="https://habr.com/manifest.webmanifest" rel="manifest">
<script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-2d0e544c.95c18ea8.js"></script><meta http-equiv="origin-trial" content="A88otRz1Fd3Nt567e2IYshC18LL3KGVXpVJW9oTCId4RYaygt23pbb4JqrbdIO/bwZPWEmRjBIRBu/bZbDR7Pg4AAABueyJvcmlnaW4iOiJodHRwczovL2ltYXNkay5nb29nbGVhcGlzLmNvbTo0NDMiLCJmZWF0dXJlIjoiVHJ1c3RUb2tlbnMiLCJleHBpcnkiOjE2MzQwODMxOTksImlzVGhpcmRQYXJ0eSI6dHJ1ZX0="><meta http-equiv="origin-trial" content="A0gCLbXCcL0R1Oc8tFPDs0G4Elz17w3zHp+Zst66+D17veE2o7fUcPsA114QtSTRqfVJLMeTSdeWOom0CcyCsgYAAAB7eyJvcmlnaW4iOiJodHRwczovL2RvdWJsZWNsaWNrLm5ldDo0NDMiLCJmZWF0dXJlIjoiVHJ1c3RUb2tlbnMiLCJleHBpcnkiOjE2MzQwODMxOTksImlzU3ViZG9tYWluIjp0cnVlLCJpc1RoaXJkUGFydHkiOnRydWV9"><meta http-equiv="origin-trial" content="A9RQ+LxFazAousxUwSCzaihJjHLO1UyjQp0teZKHl7WdbVjPDfHSKMd6D/ZI5MTjqClFycbl70EFd7cBJWXqKQEAAACBeyJvcmlnaW4iOiJodHRwczovL2dvb2dsZXRhZ3NlcnZpY2VzLmNvbTo0NDMiLCJmZWF0dXJlIjoiVHJ1c3RUb2tlbnMiLCJleHBpcnkiOjE2MzQwODMxOTksImlzU3ViZG9tYWluIjp0cnVlLCJpc1RoaXJkUGFydHkiOnRydWV9"><meta http-equiv="origin-trial" content="A6WKeWsdn1Ct+ZPqS9NCxxaiBoQ7wdTkK2/gE69Yu0gfBKJfo1gOvgkGmf5/xaIajT/RUb9AbnF1FsSZ47cCcQcAAACBeyJvcmlnaW4iOiJodHRwczovL2dvb2dsZXN5bmRpY2F0aW9uLmNvbTo0NDMiLCJmZWF0dXJlIjoiVHJ1c3RUb2tlbnMiLCJleHBpcnkiOjE2MzQwODMxOTksImlzU3ViZG9tYWluIjp0cnVlLCJpc1RoaXJkUGFydHkiOnRydWV9"><meta http-equiv="origin-trial" content="A04ZCu7yjrHgwQJK5ISHhH1DSg0qqowEay3n70KO6wV3D2Mj+OX3Kw20aSMitzgdG1xfrN7sOJV/dZIk+RvCzA4AAAB2eyJvcmlnaW4iOiJodHRwczovL2dvb2dsZS5jb206NDQzIiwiZmVhdHVyZSI6IlRydXN0VG9rZW5zIiwiZXhwaXJ5IjoxNjM0MDgzMTk5LCJpc1N1YmRvbWFpbiI6dHJ1ZSwiaXNUaGlyZFBhcnR5Ijp0cnVlfQ=="><meta http-equiv="origin-trial" content="A2YAd4xOntTGygIDjApOTtXOgVI3IWsd5OnOGq3RbRkIQwyqYWNl1JGRAcvtm6VOHDj4n07T/J19VqLuJn3MmQ8AAACWeyJvcmlnaW4iOiJodHRwczovL2RvdWJsZWNsaWNrLm5ldDo0NDMiLCJmZWF0dXJlIjoiQ29udmVyc2lvbk1lYXN1cmVtZW50IiwiZXhwaXJ5IjoxNjMxNjYzOTk5LCJpc1N1YmRvbWFpbiI6dHJ1ZSwiaXNUaGlyZFBhcnR5Ijp0cnVlLCJ1c2FnZSI6InN1YnNldCJ9"><meta http-equiv="origin-trial" content="A2c5Ux+hivdkLh/KbZUGr6f7SCR0mZrBVfPJ+/OuDVHNwiYv+Lo83b9z5qL8sod78bQl0pSLtbvRWURo+xRl7AIAAACceyJvcmlnaW4iOiJodHRwczovL2dvb2dsZXN5bmRpY2F0aW9uLmNvbTo0NDMiLCJmZWF0dXJlIjoiQ29udmVyc2lvbk1lYXN1cmVtZW50IiwiZXhwaXJ5IjoxNjMxNjYzOTk5LCJpc1N1YmRvbWFpbiI6dHJ1ZSwiaXNUaGlyZFBhcnR5Ijp0cnVlLCJ1c2FnZSI6InN1YnNldCJ9"><meta http-equiv="origin-trial" content="AzNJ4sd3tVurolpdvWYZ4cmP9Po7RJhEHSqmC3pgxW9fFVZvchhtcMUgHAs97npxMD1jhXHO8s6q6Wy1MMLxKgEAAACceyJvcmlnaW4iOiJodHRwczovL2dvb2dsZXRhZ3NlcnZpY2VzLmNvbTo0NDMiLCJmZWF0dXJlIjoiQ29udmVyc2lvbk1lYXN1cmVtZW50IiwiZXhwaXJ5IjoxNjMxNjYzOTk5LCJpc1N1YmRvbWFpbiI6dHJ1ZSwiaXNUaGlyZFBhcnR5Ijp0cnVlLCJ1c2FnZSI6InN1YnNldCJ9"><meta http-equiv="origin-trial" content="AwfG8hAcHnPa/kJ1Co0EvG/K0F9l1s2JZGiDLt2mhC3QI5Fh4qmsmSwrWObZFbRC9ieDaSLU6lHRxhGUF/i9sgoAAACBeyJvcmlnaW4iOiJodHRwczovL2RvdWJsZWNsaWNrLm5ldDo0NDMiLCJmZWF0dXJlIjoiSW50ZXJlc3RDb2hvcnRBUEkiLCJleHBpcnkiOjE2MjYyMjA3OTksImlzU3ViZG9tYWluIjp0cnVlLCJpc1RoaXJkUGFydHkiOnRydWV9"><meta http-equiv="origin-trial" content="AwQ7dCmHkvR6FuOFxAuNnktYSQrGbL4dF+eBkrwNLALc69Wr//PnO1yzns3pjUoCaYbKHtVcnng2hU+8OUm0PAYAAACHeyJvcmlnaW4iOiJodHRwczovL2dvb2dsZXN5bmRpY2F0aW9uLmNvbTo0NDMiLCJmZWF0dXJlIjoiSW50ZXJlc3RDb2hvcnRBUEkiLCJleHBpcnkiOjE2MjYyMjA3OTksImlzU3ViZG9tYWluIjp0cnVlLCJpc1RoaXJkUGFydHkiOnRydWV9"><meta http-equiv="origin-trial" content="AysVDPGQTLD/Scn78x4mLwB1tMfje5jwUpAAzGRpWsr1NzoN7MTFhT3ClmImi2svDZA7V6nWGIV8YTPsSRTe0wYAAACHeyJvcmlnaW4iOiJodHRwczovL2dvb2dsZXRhZ3NlcnZpY2VzLmNvbTo0NDMiLCJmZWF0dXJlIjoiSW50ZXJlc3RDb2hvcnRBUEkiLCJleHBpcnkiOjE2MjYyMjA3OTksImlzU3ViZG9tYWluIjp0cnVlLCJpc1RoaXJkUGFydHkiOnRydWV9"><script src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/pubads_impl_2021082501.js" async=""></script><link rel="stylesheet" type="text/css" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/page-article_sandbox_user-articles-subpages.42ab304e.css"><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/page-article_sandbox_user-articles-subpages.7c1dabd1.js"></script><link rel="stylesheet" type="text/css" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/page-article_page-company.c389b18d.css"><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/page-article_page-company.8acf483a.js"></script><link rel="stylesheet" type="text/css" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/page-article.5e02c2f9.css"><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/page-article.e4fcd1ed.js"></script><script data-vue-meta="ssr" onload="window[&#39;e0044d29c024&#39;] = true" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/saved_resource" data-vmid="checklogin"></script><script data-vue-meta="ssr" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/ads.js" onload="window[&#39;zhY4i4nJ9K&#39;] = true" data-vmid="checkad"></script><link rel="stylesheet" type="text/css" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-f458c7c4.7ecad03c.css"><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-f458c7c4.529673c8.js"></script><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-2d21ab85.a318b7bd.js"></script><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/hljs.c67ce31f.js"></script><link rel="stylesheet" type="text/css" href="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/gallery.36fc9d2c.css"><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/gallery.42a5862c.js"></script><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-2d222042.0248dfe2.js"></script><script charset="utf-8" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/photoswipe.2b48f110.js"></script><link data-vue-meta="ssr" href="https://habr.com/ru/company/ruvds/blog/328436/" rel="canonical" data-vmid="canonical"><link data-vue-meta="ssr" href="https://habr.com/ru/rss/post/328436/?fl=ru" type="application/rss+xml" title="" rel="alternate" name="rss"><link data-vue-meta="ssr" data-vmid="hreflang"><link data-vue-meta="ssr" image_src="image" href="https://habr.com/share/publication/328436/9f7ddaee48ab2f5df907918a78b4208c/" data-vmid="image:href"><meta data-vue-meta="ssr" property="og:title" content="Bash-скрипты, часть 11: expect и автоматизация интерактивных утилит" data-vmid="og:title"><meta data-vue-meta="ssr" name="twitter:title" content="Bash-скрипты, часть 11: expect и автоматизация интерактивных утилит" data-vmid="twitter:title"><meta data-vue-meta="ssr" name="aiturec:title" content="Bash-скрипты, часть 11: expect и автоматизация интерактивных утилит" data-vmid="aiturec:title"><meta data-vue-meta="ssr" name="description" content="Bash-скрипты: начало
Bash-скрипты, часть 2: циклы
Bash-скрипты, часть 3: параметры и ключи командной строки
Bash-скрипты, часть 4: ввод и вывод
Bash-скрипты, часть 5: сигналы, фоновые задачи,..." data-vmid="description"><meta data-vue-meta="ssr" itemprop="description" content="Bash-скрипты: начало
Bash-скрипты, часть 2: циклы
Bash-скрипты, часть 3: параметры и ключи командной строки
Bash-скрипты, часть 4: ввод и вывод
Bash-скрипты, часть 5: сигналы, фоновые задачи,..." data-vmid="description:itemprop"><meta data-vue-meta="ssr" property="og:description" content="Bash-скрипты: начало
Bash-скрипты, часть 2: циклы
Bash-скрипты, часть 3: параметры и ключи командной строки
Bash-скрипты, часть 4: ввод и вывод
Bash-скрипты, часть 5: сигналы, фоновые задачи,..." data-vmid="og:description"><meta data-vue-meta="ssr" name="twitter:description" content="Bash-скрипты: начало
Bash-скрипты, часть 2: циклы
Bash-скрипты, часть 3: параметры и ключи командной строки
Bash-скрипты, часть 4: ввод и вывод
Bash-скрипты, часть 5: сигналы, фоновые задачи,..." data-vmid="twitter:description"><meta data-vue-meta="ssr" property="aiturec:description" content="Bash-скрипты: начало
Bash-скрипты, часть 2: циклы
Bash-скрипты, часть 3: параметры и ключи командной строки
Bash-скрипты, часть 4: ввод и вывод
Bash-скрипты, часть 5: сигналы, фоновые задачи,..." data-vmid="aiturec:description"><meta data-vue-meta="ssr" itemprop="image" content="https://habr.com/share/publication/328436/9f7ddaee48ab2f5df907918a78b4208c/" data-vmid="image:itemprop"><meta data-vue-meta="ssr" property="og:image" content="https://habr.com/share/publication/328436/9f7ddaee48ab2f5df907918a78b4208c/" data-vmid="og:image"><meta data-vue-meta="ssr" property="aiturec:image" content="https://habr.com/share/publication/328436/9f7ddaee48ab2f5df907918a78b4208c/" data-vmid="aiturec:image"><meta data-vue-meta="ssr" name="twitter:image" content="https://habr.com/share/publication/328436/9f7ddaee48ab2f5df907918a78b4208c/" data-vmid="twitter:image"><meta data-vue-meta="ssr" property="vk:image" content="https://habr.com/share/publication/328436/9f7ddaee48ab2f5df907918a78b4208c/" data-vmid="vk:image"><meta data-vue-meta="ssr" property="aiturec:item_id" content="328436" data-vmid="aiturec:item_id"><meta data-vue-meta="ssr" property="aiturec:datetime" content="2017-05-11T12:42:16.000Z" data-vmid="aiturec:datetime"><meta data-vue-meta="ssr" property="og:type" content="article" data-vmid="og:type"><meta data-vue-meta="ssr" property="og:image:width" content="1200" data-vmid="og:image:width"><meta data-vue-meta="ssr" property="og:image:height" content="630" data-vmid="og:image:height"><script data-vue-meta="ssr" type="application/ld+json" data-vmid="ldjson-schema">{"@context":"http:\/\/schema.org","@type":"Article","mainEntityOfPage":{"@type":"WebPage","@id":"https:\/\/habr.com\/ru\/company\/ruvds\/blog\/328436\/"},"headline":"Bash-скрипты, часть 11: expect и автоматизация интерактивных утилит","datePublished":"2017-05-11T15:42:16+03:00","dateModified":"2017-05-11T15:57:17+03:00","author":{"@type":"Person","name":"ru_vds"},"publisher":{"@type":"Organization","name":"Habr","logo":{"@type":"ImageObject","url":"https:\/\/habrastorage.org\/webt\/a_\/lk\/9m\/a_lk9mjkccjox-zccjrpfolmkmq.png"}},"description":"Bash-скрипты: начало Bash-скрипты, часть 2: циклы Bash-скрипты, часть 3: параметры и ключи командной строки Bash-скрипты, часть 4: ввод и вывод Bash-скрипты, час...","url":"https:\/\/habr.com\/ru\/company\/ruvds\/blog\/328436\/#post-content-body","about":["c_ruvds","h_linux","h_sys_admin","f_admin"],"image":["https:\/\/habrastorage.org\/files\/803\/892\/bfe\/803892bfe548499aa763df324d40fd01.png","https:\/\/habrastorage.org\/files\/1ba\/550\/d25\/1ba550d25e8846ce8805de564da6aa63.png","https:\/\/habrastorage.org\/getpro\/habr\/post_images\/456\/54e\/801\/45654e80149d0ffc4f4fa6616857ef44.png","https:\/\/habrastorage.org\/getpro\/habr\/post_images\/fb8\/131\/0fe\/fb81310fe0054606d27778474d80079f.png","https:\/\/habrastorage.org\/getpro\/habr\/post_images\/10f\/e73\/1bd\/10fe731bd35562c114a5158121450126.png","https:\/\/habrastorage.org\/getpro\/habr\/post_images\/0a2\/0d5\/456\/0a20d545683e4b5b5469495a642671ab.png","https:\/\/habrastorage.org\/getpro\/habr\/post_images\/850\/bb0\/9e3\/850bb09e300047626407ed2b0c8ccee1.png","https:\/\/habrastorage.org\/getpro\/habr\/post_images\/cce\/aeb\/72a\/cceaeb72a9cfe0513d09353f1e01c485.png","https:\/\/habrastorage.org\/getpro\/habr\/post_images\/8c6\/9e0\/558\/8c69e055846d656af2cc6c11da767b73.png","https:\/\/lh6.googleusercontent.com\/19F2r17sDNowQV52cowtXGsMOl__tiRLfkzmj7MjygsMzuDL5RPeUu_Mhb2a2zYWrsfTT6ycbrAIKjtq0EPHhw58sTJqAo6HqZd5Vh8QCp15saZVlCHHZdPNoiiRxXU1w4bJXgAWpw_vIYb0BA","https:\/\/lh3.googleusercontent.com\/h3Hyo_lzUuZJnYZ9taZ1XvIY-IhWkG8eyyi8nEbELzQDP_M42RCOH2QPn1T-Qv41NMv9Zzk4O0vQUtW_SUy4Pset1Sl_wH14_YgJvYWl_ecMsl3wFzqkQIeHr_n3GU_aF6jTZqsgQhQ-wYyamw","https:\/\/lh5.googleusercontent.com\/4rvWSMdIyQR0zXItxldVlUntwL36cCK0iQ4jOKbpE_YeD123l5-DhUCZbh10y4hELKAhAuc5mfz17ZxVBzGZz0Cvg3dQnpdn8NhaollSGPua6nBHFp_6LjEXSXYl6yBu521Eg-CuHoSPDcK_fw","https:\/\/lh3.googleusercontent.com\/QzISHnYVurZOqWM7NCNcVW8-248AGrBMf0qwzxf2WijbRWqI8eVOKcWECEod_AXg551sIKSnHJR-U5sz-qHHOk5evCg48H4kadYk0Kngk16BQSU0vYhML7xsb0TG-8R_9yABDjzws0YMbesWAQ","https:\/\/habrastorage.org\/files\/1ba\/550\/d25\/1ba550d25e8846ce8805de564da6aa63.png"]}</script></head>
<body cz-shortcut-listen="true">


<div id="app" data-async-called="true"><div class="tm-layout__wrapper"><!----><div></div><!----><header class="tm-header"><div class="tm-page-width"><div class="tm-header__container"><!----><span class="tm-header__logo-wrap"><a href="https://habr.com/ru/" class="tm-header__logo tm-header__logo_ru"><svg height="16" width="16" class="tm-svg-img tm-header__icon"><title>Хабр</title><use xlink:href="/img/habr-logo-ru.svg#logo"></use></svg></a><span class="tm-header__beta-sign" style="display: none;">β</span></span><div class="tm-dropdown tm-header__projects"><div class="tm-dropdown__head"><button class="tm-header__dropdown-toggle"><svg height="16" width="16" class="tm-svg-img tm-header__icon tm-header__icon_dropdown"><title>Открыть список</title><use xlink:href="/img/megazord-v24.cee85629.svg#arrow-down"></use></svg></button></div><!----></div><a href="https://habr.com/ru/sandbox/start/" class="tm-header__become-author-btn">
              Как стать автором
            </a><!----><!----><div class="tm-header-user-menu tm-header_user-menu"><a href="https://habr.com/ru/search/" class="tm-header-user-menu__item tm-header-user-menu__search"><svg height="24" width="24" class="tm-svg-img tm-header-user-menu__icon tm-header-user-menu__icon_search"><title>Поиск</title><use xlink:href="/img/megazord-v24.cee85629.svg#search"></use></svg></a><!----><!----><!----><div class="tm-header-user-menu__item"><button data-test-id="menu-toggle-guest" class="tm-header-user-menu__toggle"><svg height="24" width="24" class="tm-svg-img tm-header-user-menu__icon tm-header-user-menu__icon_white"><title>Профиль</title><use xlink:href="/img/megazord-v24.cee85629.svg#header-user"></use></svg></button><!----></div><!----></div></div></div></header><div class="tm-layout"><div class="tm-page-progress-bar"></div><div class="tm-base-layout__header"><div class="tm-page-width"><div class="tm-base-layout__header-wrapper"><div class="tm-main-menu"><div class="tm-main-menu__section"><nav class="tm-main-menu__section-content"><!----><a href="https://habr.com/ru/all/" class="tm-main-menu__item">
        Все потоки
      </a><a href="https://habr.com/ru/flows/develop/" class="tm-main-menu__item">
          Разработка
        </a><a href="https://habr.com/ru/flows/admin/" class="tm-main-menu__item">
          Администрирование
        </a><a href="https://habr.com/ru/flows/design/" class="tm-main-menu__item">
          Дизайн
        </a><a href="https://habr.com/ru/flows/management/" class="tm-main-menu__item">
          Менеджмент
        </a><a href="https://habr.com/ru/flows/marketing/" class="tm-main-menu__item">
          Маркетинг
        </a><a href="https://habr.com/ru/flows/popsci/" class="tm-main-menu__item">
          Научпоп
        </a></nav></div></div><!----></div></div></div><!----><div class="tm-page-width tm-page-width_no-padding"></div><main class="tm-layout__container"><div class="tm-page" hl="ru" companyname="ruvds"><div class="tm-page-width tm-page-width_no-padding"><div class="tm-page__header"><div class="tm-company-card__branding"><!----><a href="https://ruvds.com/ru-rub"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/b778d5a41f83016379c6630262c3294c.png" width="100%" class="tm-company-card__branding-image"></a></div></div><div class="tm-page__wrapper"><div class="tm-page__main tm-page__main_has-sidebar"><div class="pull-down"><div class="pull-down__header" style="height: 0px;"><div class="pull-down__content" style="bottom: 10px;"><svg height="24" width="24" class="tm-svg-img pull-down__arrow"><title>Обновить</title><use xlink:href="/img/megazord-v24.cee85629.svg#pull-arrow"></use></svg></div></div><!----><div class="tm-page-article__body"><article class="tm-page-article__content tm-page-article__content_inner"><div class="tm-page-article__head-wrapper"><!----><div class="tm-article-snippet tm-page-article__snippet"><div class="tm-article-snippet__meta-container"><div class="tm-article-snippet__meta"><span class="tm-user-info tm-article-snippet__author"><a href="https://habr.com/ru/users/ru_vds/" class="tm-user-info__userpic" title="ru_vds"><div class="tm-entity-image"><img alt="" height="24" loading="lazy" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/e36812d3de38eb3d479d7a13228717ef.png" width="24" class="tm-entity-image__pic"></div></a><span class="tm-user-info__user"><a href="https://habr.com/ru/users/ru_vds/" class="tm-user-info__username">
      ru_vds
    </a></span></span><span class="tm-article-snippet__datetime-published"><time datetime="2017-05-11T12:42:16.000Z" title="2017-05-11, 19:42">11  мая  2017 в 19:42</time></span></div><!----></div><h1 lang="ru" class="tm-article-snippet__title tm-article-snippet__title_h1"><span>Bash-скрипты, часть 11: expect и автоматизация интерактивных утилит</span></h1><div class="tm-article-snippet__hubs"><span class="tm-article-snippet__hubs-item"><a href="https://habr.com/ru/company/ruvds/blog/" class="tm-article-snippet__hubs-item-link router-link-active"><span>Блог компании RUVDS.com</span><!----></a></span><span class="tm-article-snippet__hubs-item"><a href="https://habr.com/ru/hub/linux/" class="tm-article-snippet__hubs-item-link"><span>Настройка Linux</span><span title="Профильный хаб" class="tm-article-snippet__profiled-hub">*</span></a></span><span class="tm-article-snippet__hubs-item"><a href="https://habr.com/ru/hub/sys_admin/" class="tm-article-snippet__hubs-item-link"><span>Системное администрирование</span><span title="Профильный хаб" class="tm-article-snippet__profiled-hub">*</span></a></span></div><div class="tm-article-snippet__labels"><div class="tm-article-snippet__label"><span>
          Перевод
        </span></div></div><!----><!----></div></div><div class="tm-page-article__origin"><a href="https://likegeeks.com/expect-command/" target="_blank" class="tm-page-article__origin-link">
                Автор оригинала:
                <span class="tm-page-article__origin-value">
                  Администратор likegeeks.com
                </span></a></div><div data-gallery-root="" class="tm-article-body" lang="ru"><div class="article-formatted-body article-formatted-body_version-1" id="post-content-body"><div xmlns="http://www.w3.org/1999/xhtml"><blockquote><a href="https://habrahabr.ru/company/ruvds/blog/325522/">Bash-скрипты: начало</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/325928/">Bash-скрипты, часть 2: циклы</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/326328/">Bash-скрипты, часть 3: параметры и ключи командной строки</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/326594/">Bash-скрипты, часть 4: ввод и вывод</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/326826/">Bash-скрипты, часть 5: сигналы, фоновые задачи, управление сценариями</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/327248/">Bash-скрипты, часть 6: функции и разработка библиотек</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/327530/">Bash-скрипты, часть 7: sed и обработка текстов</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/327754/">Bash-скрипты, часть 8: язык обработки данных awk</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/327896/">Bash-скрипты, часть 9: регулярные выражения</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/328346/">Bash-скрипты, часть 10: практические примеры</a><br>
<a href="https://habrahabr.ru/company/ruvds/blog/328436/">Bash-скрипты, часть 11: expect и автоматизация интерактивных утилит</a></blockquote><br>
В прошлый раз мы говорили о методике разработки bash-скриптов. Если же суммировать всё, что мы разобрали в предыдущих десяти материалах, то вы, если начинали читать их, ничего не зная о bash, теперь можете сделать уже довольно много всего полезного.<br>
<br>
<a href="https://habrahabr.ru/company/ruvds/blog/328436/"><img data-src="https://habrastorage.org/files/803/892/bfe/803892bfe548499aa763df324d40fd01.png" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/803892bfe548499aa763df324d40fd01.png"></a><br>
<br>
Сегодняшняя тема, заключительная в этой серии материалов, посвящена автоматизации работы с интерактивными утилитами, например, со скриптами, которые, в процессе выполнения, взаимодействуют с пользователем. В этом деле нам поможет expect — инструмент, основанный на языке Tcl.<br>
<a name="habracut"></a><br>
Expect позволяет создавать программы, ожидающие вопросов от других программ и дающие им ответы. Expect можно сравнить с роботом, который способен заменить пользователя при взаимодействии со сценариями командной строки.<br>
<br>
<a href="https://ruvds.com/ru-rub/#order"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/files/1ba/550/d25/1ba550d25e8846ce8805de564da6aa63.png"></a><br>
<br>
<h2><font color="#3AC1EF">Основы expect</font></h2><br>
Если expect в вашей системе не установлен, исправить это, например, в Ubuntu, можно так:<br>
<br>
<pre><code class="hljs kotlin">$ apt-<span class="hljs-keyword">get</span> install <span class="hljs-keyword">expect</span></code></pre><br>
В чём-то вроде CentOs установка выполняется такой командой:<br>
<br>
<pre><code class="hljs cmake">$ yum <span class="hljs-keyword">install</span> expect</code></pre><br>
Expect предоставляет набор команд, позволяющих взаимодействовать с утилитами командной строки. Вот его основные команды:<br>
<br>
<ul>
<li><code>spawn — </code>запуск процесса или программы. Например, это может быть командная оболочка, <a href="https://likegeeks.com/ftp-server-linux/">FTP</a>, Telnet, ssh, scp и так далее.<br>
 <br>
</li>
<li><code>expect — </code>ожидание данных, выводимых программой. При написании скрипта можно указать, какого именно вывода он ждёт и как на него нужно реагировать.<br>
<br>
</li>
<li><code>send — </code>отправка ответа. Expect-скрипт с помощью этой команды может отправлять входные данные автоматизируемой программе. Она похожа на знакомую вам команду <code>echo</code> в обычных bash-скриптах.<br>
<br>
</li>
<li><code>interact — </code>позволяет переключиться на «ручной» режим управления программой.<br>
</li>
</ul><br>
<h2><font color="#3AC1EF">Автоматизация bash-скрипта</font></h2><br>
Напишем скрипт, который взаимодействует с пользователем и автоматизируем его с помощью expect. Вот код bash-скрипта <code>questions</code>:<br>
<br>
<pre><code class="hljs bash"><span class="hljs-meta">#!/bin/bash</span>
<span class="hljs-built_in">echo</span> <span class="hljs-string">"Hello, who are you?"</span>
<span class="hljs-built_in">read</span> <span class="hljs-variable">$REPLY</span>
<span class="hljs-built_in">echo</span> <span class="hljs-string">"Can I ask you some questions?"</span>
<span class="hljs-built_in">read</span> <span class="hljs-variable">$REPLY</span>
<span class="hljs-built_in">echo</span> <span class="hljs-string">"What is your favorite topic?"</span>
<span class="hljs-built_in">read</span> <span class="hljs-variable">$REPLY</span></code></pre><br>
Теперь напишем expect-скрипт, который запустит скрипт <code>questions</code> и будет отвечать на его вопросы:<br>
<br>
<pre><code class="hljs kotlin">#!/usr/bin/<span class="hljs-keyword">expect</span> -f
<span class="hljs-keyword">set</span> timeout -<span class="hljs-number">1</span>
spawn ./questions
<span class="hljs-keyword">expect</span> <span class="hljs-string">"Hello, who are you?\r"</span>
send -- <span class="hljs-string">"Im Adam\r"</span>
<span class="hljs-keyword">expect</span> <span class="hljs-string">"Can I ask you some questions?\r"</span>
send -- <span class="hljs-string">"Sure\r"</span>
<span class="hljs-keyword">expect</span> <span class="hljs-string">"What is your favorite topic?\r"</span>
send -- <span class="hljs-string">"Technology\r"</span>
<span class="hljs-keyword">expect</span> eof</code></pre><br>
Сохраним скрипт, дав ему имя <code>answerbot</code>.<br>
<br>
В начале скрипта находится строка идентификации, которая, в данном случае, содержит путь к expect, так как интерпретировать скрипт будет именно expect.<br>
Во второй строке мы отключаем тайм-аут, устанавливая переменную expect <code>timeout</code> в значение -1. Остальной код — это и есть автоматизация работы с bash-скриптом.<br>
<br>
Сначала, с помощью команды <code>spawn</code>, мы запускаем bash-скрипт. Естественно, тут может быть вызвана любая другая утилита командной строки. Далее задана последовательность вопросов, поступающих от bash-скрипта, и ответов, которые даёт на них expect. Получив вопрос от подпроцесса, expect выдаёт ему заданный ответ и ожидает следующего вопроса.<br>
<br>
В последней команде expect ожидает признака конца файла, скрипт, дойдя до этой команды, завершается.<br>
<br>
Теперь пришло время всё это опробовать. Сделаем <code>answerbot</code> исполняемым файлом:<br>
<br>
<pre><code class="hljs perl">$ <span class="hljs-keyword">chmod</span> +<span class="hljs-keyword">x</span> ./answerbot</code></pre><br>
И вызовем его:<br>
<br>
<pre><code class="hljs javascript">$./answerbot</code></pre><br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/getpro/habr/post_images/456/54e/801/45654e80149d0ffc4f4fa6616857ef44.png"></div><br>
<i><font color="#999999">Expect-скрипт отвечает на вопросы bash-скрипта</font></i><br>
<br>
Как видно, expect-скрипт верно ответил на вопросы bash-скрипта. Если на данном этапе вы столкнулись с ошибкой, вызванной тем, что неправильно указано расположение expect, выяснить его адрес можно так:<br>
<br>
<pre><code class="hljs bash">$ <span class="hljs-built_in">which</span> expect</code></pre><br>
Обратите внимание на то, что после запуска скрипта <code>answerbot</code> всё происходит в полностью автоматическом режиме. То же самое можно проделать для любой утилиты командной строки. Тут надо отметить, что наш bash-скрипт устроен очень просто, мы точно знаем, какие именно данные он выводит, поэтому написать expect-скрипт для взаимодействия с ним несложно. Задача усложняется при работе с программами, которые написаны другими разработчиками. Однако, здесь на помощь приходит средство для автоматизированного создания expect-скриптов.<br>
<br>
<h2><font color="#3AC1EF">Autoexpect — автоматизированное создание expect-скриптов</font></h2><br>
Autoexpect позволяет запускать программы, которые надо автоматизировать, после чего записывает то, что они выводят, и то, что пользователь вводит, отвечая на их вопросы. Вызовем autoexpect, передав этой утилите имя нашего скрипта:<br>
<br>
<pre><code class="hljs powershell"><span class="hljs-variable">$</span> autoexpect ./questions</code></pre><br>
В этом режиме взаимодействие с bash-скриптом ничем не отличается от обычного: мы сами вводим ответы на его вопросы.<br>
<br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/getpro/habr/post_images/fb8/131/0fe/fb81310fe0054606d27778474d80079f.png"></div><br>
<i><font color="#999999">Запуск bash-скрипта с помощью autoexpect</font></i><br>
<br>
После завершения работы с bash-скриптом, autoexpect сообщит о том, что собранные данные записаны в файл <code>script.exp</code>. Взглянем на этот файл.<br>
<br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/getpro/habr/post_images/10f/e73/1bd/10fe731bd35562c114a5158121450126.png"></div><br>
<i><font color="#999999">Файл script.exp</font></i><br>
<br>
В целом, за исключением некоторых деталей, перед нами такой же скрипт, который мы писали самостоятельно. Если запустить этот скрипт, результат будет тем же.<br>
<br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/getpro/habr/post_images/0a2/0d5/456/0a20d545683e4b5b5469495a642671ab.png"></div><br>
<i><font color="#999999">Запуск expect-скрипта, созданного автоматически</font></i><br>
<br>
При записи сеансов взаимодействия с некоторыми программами, вроде FTP-клиентов, вы можете столкнуться с тем, что они используют в выводимых данных сведения о времени проведения операции, или выводят данные, отражающие процесс выполнения неких продолжительных действий. В целом, речь идёт о том, что вывод программы при каждом её запуске, правильно воспринимаемый человеком и вызывающий ввод одних и тех же ответов, будет, в тех же условиях, выглядеть по-новому для expect.<br>
<br>
Если в expect-скрипте строки, ожидаемые от такой программы, будут жёстко зафиксированы, такой скрипт не сможет нормально работать. Справиться с этим можно, либо удалив из expect-скрипта данные, которые выглядят по-новому при каждом запуске программы, либо использовав шаблоны, пользуясь которыми, expect сможет правильно понять то, что хочет от него программа.<br>
<br>
Как видите, autoexpect — это весьма полезный инструмент, но и он не лишён недостатков, исправить которые можно только вручную. Поэтому продолжим осваивать язык expect-скриптов.<br>
<br>
<h2><font color="#3AC1EF">Работа с переменными и параметрами командной строки</font></h2><br>
Для объявления переменных в expect-скриптах используется команда <code>set</code>. Например, для того, чтобы присвоить значение 5 переменной <code>VAR1</code>, используется следующая конструкция:<br>
<br>
<pre><code class="hljs actionscript"><span class="hljs-keyword">set</span> VAR1 <span class="hljs-number">5</span></code></pre><br>
Для доступа к значению переменной перед её именем надо добавить знак доллара — <code>$</code>. В нашем случае это будет выглядеть как <code>$VAR1</code>.<br>
<br>
Для того, чтобы получить доступ к аргументам командной строки, с которыми вызван expect-скрипт, можно поступить так:<br>
<br>
<pre><code class="hljs bash"><span class="hljs-built_in">set</span> VAR [lindex <span class="hljs-variable">$argv</span> 0]</code></pre><br>
Тут мы объявляем переменную <code>VAR</code> и записываем в неё указатель на первый аргумент командной строки, <code>$argv 0</code>.<br>
<br>
Для целей обновлённого expect-скрипта мы собираемся записать значение первого аргумента, представляющее собой имя пользователя, которое будет использовано в программе, в переменную <code>my_name</code>. Второй аргумент, символизирующий то, что пользователю нравится, попадёт в переменную <code>my_favorite</code>. В результате объявление переменных будет выглядеть так:<br>
<br>
<pre><code class="hljs bash"><span class="hljs-built_in">set</span> my_name [lindex <span class="hljs-variable">$argv</span> 0]
<span class="hljs-built_in">set</span> my_favorite [lindex <span class="hljs-variable">$argv</span> 1]</code></pre><br>
Отредактируем скрипт <code>answerbot</code>, приведя его к такому виду:<br>
<br>
<pre><code class="hljs kotlin">#!/usr/bin/<span class="hljs-keyword">expect</span> -f
<span class="hljs-keyword">set</span> my_name [lindex $argv <span class="hljs-number">0</span>]
<span class="hljs-keyword">set</span> my_favorite [lindex $argv <span class="hljs-number">1</span>]
<span class="hljs-keyword">set</span> timeout -<span class="hljs-number">1</span>
spawn ./questions
<span class="hljs-keyword">expect</span> <span class="hljs-string">"Hello, who are you?\r"</span>
send -- <span class="hljs-string">"Im <span class="hljs-variable">$my_name</span>\r"</span>
<span class="hljs-keyword">expect</span> <span class="hljs-string">"Can I ask you some questions?\r"</span>
send -- <span class="hljs-string">"Sure\r"</span>
<span class="hljs-keyword">expect</span> <span class="hljs-string">"What is your favorite topic?\r"</span>
send -- <span class="hljs-string">"<span class="hljs-variable">$my_favorite</span>\r"</span>
<span class="hljs-keyword">expect</span> eof</code></pre><br>
Запустим его, передав в качестве первого параметра <code>SomeName</code>, в качестве второго — <code>Programming</code>:<br>
<br>
<pre><code class="hljs powershell"><span class="hljs-variable">$</span> ./answerbot SomeName Programming</code></pre><br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/getpro/habr/post_images/850/bb0/9e3/850bb09e300047626407ed2b0c8ccee1.png"></div><br>
<i><font color="#999999">Expect-скрипт, использующий переменные и параметры командной строки</font></i><br>
<br>
Как видите, всё работает так, как ожидалось. Теперь expect-скрипт отвечает на вопросы bash-скрипта, пользуясь переданными ему параметрами командной строки.<br>
<br>
<h2><font color="#3AC1EF">Ответы на разные вопросы, которые могут появиться в одном и том же месте</font></h2><br>
Если автоматизируемая программа может, в одной ситуации, выдать одну строку, а в другой, в том же самом месте — другую, в expect можно использовать блоки, заключённые в фигурные скобки и содержащие варианты реакции скрипта на разные данные, полученные от программы. Выглядит это так:<br>
<br>
<pre><code class="hljs perl">expect {
 &nbsp;&nbsp;&nbsp;<span class="hljs-string">"something"</span> { <span class="hljs-keyword">send</span> -- <span class="hljs-string">"send this\r"</span> }
 &nbsp;&nbsp;&nbsp;<span class="hljs-string">"*another"</span> { <span class="hljs-keyword">send</span> -- <span class="hljs-string">"send another\r"</span> }
}</code></pre><br>
Здесь, если expect-скрипт увидит строку «something», он отправит ответ «send this». Если же это будет некая строка, оканчивающаяся на «another», он отправит ответ «send another».<br>
<br>
Напишем новый скрипт, записав его в файл <code>questions</code>, случайным образом задающий в одном и том же месте разные вопросы:<br>
<br>
<pre><code class="hljs bash"><span class="hljs-meta">#!/bin/bash</span>
<span class="hljs-built_in">let</span> number=<span class="hljs-variable">$RANDOM</span>
<span class="hljs-keyword">if</span> [ <span class="hljs-variable">$number</span> -gt 25000 ]
<span class="hljs-keyword">then</span>
<span class="hljs-built_in">echo</span> <span class="hljs-string">"What is your favorite topic?"</span>
<span class="hljs-keyword">else</span>
<span class="hljs-built_in">echo</span> <span class="hljs-string">"What is your favorite movie?"</span>
<span class="hljs-keyword">fi</span>
<span class="hljs-built_in">read</span> <span class="hljs-variable">$REPLY</span></code></pre><br>
Тут мы генерируем случайное число при каждом запуске скрипта, и, проанализировав его, выводим один из двух вопросов.<br>
<br>
Для автоматизации такого скрипта нам и пригодится вышеописанная конструкция:<br>
<br>
<pre><code class="hljs javascript"><span class="hljs-meta">#!/usr/bin/expect -f</span>
<span class="hljs-keyword">set</span> timeout -1
spawn ./questions
expect {
 &nbsp;&nbsp;&nbsp;<span class="hljs-string">"*topic?"</span> { send -- <span class="hljs-string">"Programming\r"</span> }
 &nbsp;&nbsp;&nbsp;<span class="hljs-string">"*movie?"</span> { send -- <span class="hljs-string">"Star wars\r"</span> }
}</code></pre><br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/getpro/habr/post_images/cce/aeb/72a/cceaeb72a9cfe0513d09353f1e01c485.png"></div><br>
<i><font color="#999999">Ответы на разные вопросы, появляющиеся в одном и том же месте</font></i><br>
<br>
Как видно, когда автоматизированный скрипт выводит строку, оканчивающуюся на «topic?», expect-скрипт передаёт ему строку «Programming». Получив в том же месте, при другом запуске программы, вопрос, оканчивающийся на «movie?», expect-скрипт отвечает: «Star wars». Это очень полезная техника.<br>
<br>
<h2><font color="#3AC1EF">Условный оператор</font></h2><br>
Expect поддерживает условный оператор <code>if-else</code> и другие управляющие конструкции. Вот пример использования условного оператора:<br>
<br>
<pre><code class="hljs bash"><span class="hljs-comment">#!/usr/bin/expect -f</span>
<span class="hljs-built_in">set</span> TOTAL 1
<span class="hljs-keyword">if</span> { <span class="hljs-variable">$TOTAL</span> &lt; 5 } {
puts <span class="hljs-string">"\nTOTAL is less than 5\n"</span>
} elseif { <span class="hljs-variable">$TOTAL</span> &gt; 5 } {
puts <span class="hljs-string">"\nTOTAL greater than 5\n"</span>
} <span class="hljs-keyword">else</span> {
puts <span class="hljs-string">"\nTOTAL is equal to 5\n"</span>
}
expect eof</code></pre><br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/getpro/habr/post_images/8c6/9e0/558/8c69e055846d656af2cc6c11da767b73.png"></div><br>
<i><font color="#999999">Условный оператор в expect</font></i><br>
<br>
Тут мы присваиваем переменной <code>TOTAL</code> некое число, после чего проверяем его и выводим текст, зависящий от результата проверки.<br>
<br>
Обратите внимание на конфигурацию фигурных скобок. Очередная открывающая скобка должна быть расположена на той же строке, что и предыдущие конструкции.<br>
<br>
<h2><font color="#3AC1EF">Цикл while</font></h2><br>
Циклы <code>while</code> в expect очень похожи на те, что используются в обычных bash-скриптах, но, опять же, тут применяются фигурные скобки:<br>
<br>
<pre><code class="hljs sql"><span class="hljs-comment">#!/usr/bin/expect -f</span>
<span class="hljs-keyword">set</span> <span class="hljs-keyword">COUNT</span> <span class="hljs-number">0</span>
<span class="hljs-keyword">while</span> { $<span class="hljs-keyword">COUNT</span> &lt;= <span class="hljs-number">5</span> } {
puts <span class="hljs-string">"\nCOUNT is currently at $COUNT"</span>
<span class="hljs-keyword">set</span> <span class="hljs-keyword">COUNT</span> [ expr $<span class="hljs-keyword">COUNT</span> + <span class="hljs-number">1</span> ]
}
puts <span class="hljs-string">""</span></code></pre><br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image7.png"></div><br>
<i><font color="#999999">Цикл while в expect</font></i><br>
 <br>
<h2><font color="#3AC1EF">Цикл for</font></h2><br>
Цикл <code>for</code> в expect устроен по-особому. В начале цикла, в самостоятельных парах фигурных скобок, надо указать переменную-счётчик, условие прекращения цикла и правило модификации счётчика. Затем, опять же в фигурных скобках, идёт тело цикла:<br>
<br>
<pre><code class="hljs bash"><span class="hljs-comment">#!/usr/bin/expect -f</span>
<span class="hljs-keyword">for</span> {<span class="hljs-built_in">set</span> COUNT 0} {<span class="hljs-variable">$COUNT</span> &lt;= 5} {incr COUNT} {
puts <span class="hljs-string">"\nCOUNT is at <span class="hljs-variable">$COUNT</span>"</span>
}
puts <span class="hljs-string">""</span></code></pre><br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image9.png"></div><br>
<i><font color="#999999">Цикл for в expect</font></i><br>
 <br>
<h2><font color="#3AC1EF">Объявление и использование функций</font></h2><br>
Expect позволяет программисту объявлять функции, используя ключевое слово <code>proc</code>:<br>
<br>
<pre><code class="hljs nginx"><span class="hljs-attribute">proc</span> myfunc { <span class="hljs-attribute">MY_COUNT</span> } {
<span class="hljs-attribute">set</span> MY_COUNT [expr <span class="hljs-variable">$MY_COUNT</span> + <span class="hljs-number">1</span>]
return <span class="hljs-string">"<span class="hljs-variable">$MY_COUNT</span>"</span>
}</code></pre><br>
Вот как выглядит expect-скрипт, в котором используется объявленная в нём же функция:<br>
<br>
<pre><code class="hljs bash"><span class="hljs-comment">#!/usr/bin/expect -f</span>

proc myfunc { MY_COUNT } {
<span class="hljs-built_in">set</span> MY_COUNT [expr <span class="hljs-variable">$MY_COUNT</span> + 1]
<span class="hljs-built_in">return</span> <span class="hljs-string">"<span class="hljs-variable">$MY_COUNT</span>"</span>
}

<span class="hljs-built_in">set</span> COUNT 0
<span class="hljs-keyword">while</span> {<span class="hljs-variable">$COUNT</span> &lt;= 5} {
puts <span class="hljs-string">"\nCOUNT is currently at <span class="hljs-variable">$COUNT</span>"</span>
<span class="hljs-built_in">set</span> COUNT [myfunc <span class="hljs-variable">$COUNT</span>]
}
puts <span class="hljs-string">""</span></code></pre><br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image10.png"></div><br>
<i><font color="#999999">Функции в expect</font></i><br>
<br>
<h2><font color="#3AC1EF">Команда interact</font></h2><br>
Случается так, что автоматизируемые с помощью expect программы требуют ввода конфиденциальных данных, вроде паролей, которые вам не хотелось бы хранить в виде обычного текста в коде скрипта. В подобной ситуации можно воспользоваться командой <code>interact</code>, которая позволит вам, автоматизировав некую часть взаимодействия с программой, самостоятельно ввести, скажем, пароль, а потом опять передать управление expect.<br>
<br>
Когда выполняется эта команда, expect-скрипт переключается на чтение ответа на вопрос программы с клавиатуры, вместо того, чтобы передавать ей ранее записанные в нём данные.<br>
<br>
Вот bash-скрипт, в общем-то, точно такой же, как мы рассматривали ранее, но теперь ожидающий ввод пароля в ответ на один из своих вопросов:<br>
<br>
<pre><code class="hljs bash"><span class="hljs-meta">#!/bin/bash</span>
<span class="hljs-built_in">echo</span> <span class="hljs-string">"Hello, who are you?"</span>
<span class="hljs-built_in">read</span> <span class="hljs-variable">$REPLY</span>
<span class="hljs-built_in">echo</span> <span class="hljs-string">"What is you password?"</span>
<span class="hljs-built_in">read</span> <span class="hljs-variable">$REPLY</span>
<span class="hljs-built_in">echo</span> <span class="hljs-string">"What is your favorite topic?"</span>
<span class="hljs-built_in">read</span> <span class="hljs-variable">$REPLY</span></code></pre><br>
Напишем expect-скрипт, который, когда ему предлагают предоставить пароль, передаёт управление нам:<br>
<br>
<pre><code class="hljs kotlin">#!/usr/bin/<span class="hljs-keyword">expect</span> -f
<span class="hljs-keyword">set</span> timeout -<span class="hljs-number">1</span>
spawn ./questions
<span class="hljs-keyword">expect</span> <span class="hljs-string">"Hello, who are you?\r"</span>
send -- <span class="hljs-string">"Hi Im Adam\r"</span>
<span class="hljs-keyword">expect</span> <span class="hljs-string">"*password?\r"</span>
interact ++ <span class="hljs-keyword">return</span>
send <span class="hljs-string">"\r"</span>
<span class="hljs-keyword">expect</span> <span class="hljs-string">"*topic?\r"</span>
send -- <span class="hljs-string">"Technology\r"</span>
<span class="hljs-keyword">expect</span> eof</code></pre><br>
<div style="text-align:center;"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image11.png"></div><br>
<i><font color="#999999">Команда interact в expect-скрипте</font></i><br>
<br>
Встретив команду <code>interact</code>, expect-скрипт остановится, предоставив нам возможность ввести пароль. После ввода пароля надо ввести «++» и expect-скрипт продолжит работу, снова получив управление.<br>
<br>
<h2><font color="#3AC1EF">Итоги</font></h2><br>
Возможностями expect можно пользоваться в программах, написанных на разных языках программирования благодаря соответствующим библиотекам. Среди этих языков — C#, Java, Perl, Python, Ruby, и другие. То, что expect доступен для разных сред разработки — далеко не случайность. Всё дело в том, что это действительно важный и полезный инструмент, который используют для решения множества задач. Здесь и проверка качества ПО, и выполнение различных работ по сетевому администрированию, автоматизация передачи файлов, автоматическая установка обновлений и многое другое.<br>
<br>
Освоив этот материал, вы ознакомились с основными концепциями expect и научились пользоваться инструментом autoexpect для автоматического формирования скриптов. Теперь вы вполне можете продолжить изучение expect, воспользовавшись дополнительными источниками. Вот — <a href="http://wiki.tcl.tk/11584">сборник</a> учебных и справочных материалов. Вот — достойная внимания серия из трёх статей (<a href="https://www.ibm.com/developerworks/ru/library/l-expect_1/">1</a>, <a href="https://www.ibm.com/developerworks/ru/library/l-expect_2/">2</a>, <a href="https://www.ibm.com/developerworks/ru/library/l-expect_3/">3</a>). А вот — <a href="https://www.nist.gov/services-resources/software/expect">официальная страница</a> expect, на которой можно найти ссылки на исходный код программы и список публикаций.<br>
<br>
На этом мы завершаем серию материалов о bash-скриптах. Надеемся, её одиннадцать частей, а также бессчётное число комментариев к ним, помогли в достижении цели тем, кто хотел научиться писать сценарии командной строки.<br>
<br>
<a href="https://ruvds.com/ru-rub/#order"><img src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/image-loader.svg" data-src="https://habrastorage.org/files/1ba/550/d25/1ba550d25e8846ce8805de564da6aa63.png"></a><br>
<br>
Уважаемые читатели! Благодарим всех, кто был с нами. Желаем удачной автоматизации работы в Linux! И, если у вас есть опыт работы с expect — ждём ваших рассказов.</div></div><div aria-hidden="true" role="dialog" tabindex="-1" class="pswp"><div class="pswp__bg"></div><div class="pswp__scroll-wrap"><div class="pswp__container"><div class="pswp__item"></div><div class="pswp__item"></div><div class="pswp__item"></div></div><div class="pswp__ui pswp__ui--hidden"><div class="pswp__top-bar"><span class="tm-svg-icon__wrapper pswp__button pswp__button--close"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Закрыть</title><use xlink:href="/img/megazord-v24.cee85629.svg#close"></use></svg></span><div class="pswp__preloader"><div class="pswp__preloader__icn"><div class="pswp__preloader__cut"><div class="pswp__preloader__donut"></div></div></div></div></div></div></div></div><!----></div><div class="tm-article-body__tags"><div class="tm-article-body__tags-links"><span class="tm-article-body__tags-title">Теги:</span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/search/?target_type=posts&amp;order=relevance&amp;q=%5BLinux%5D" class="tm-article-body__tags-item-link">Linux</a></span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/search/?target_type=posts&amp;order=relevance&amp;q=%5BBash%5D" class="tm-article-body__tags-item-link">Bash</a></span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/search/?target_type=posts&amp;order=relevance&amp;q=%5B%D1%81%D1%86%D0%B5%D0%BD%D0%B0%D1%80%D0%B8%D0%B9%20%D0%BA%D0%BE%D0%BC%D0%B0%D0%BD%D0%B4%D0%BD%D0%BE%D0%B9%20%D1%81%D1%82%D1%80%D0%BE%D0%BA%D0%B8%5D" class="tm-article-body__tags-item-link">сценарий командной строки</a></span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/search/?target_type=posts&amp;order=relevance&amp;q=%5BBash-%D1%81%D0%BA%D1%80%D0%B8%D0%BF%D1%82%5D" class="tm-article-body__tags-item-link">Bash-скрипт</a></span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/search/?target_type=posts&amp;order=relevance&amp;q=%5B%D0%B0%D0%B2%D1%82%D0%BE%D0%BC%D0%B0%D1%82%D0%B8%D0%B7%D0%B0%D1%86%D0%B8%D1%8F%5D" class="tm-article-body__tags-item-link">автоматизация</a></span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/search/?target_type=posts&amp;order=relevance&amp;q=%5Bexpect%5D" class="tm-article-body__tags-item-link">expect</a></span></div><div class="tm-article-body__tags-links"><span class="tm-article-body__tags-title">Хабы:</span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/company/ruvds/blog/" class="tm-article-body__tags-item-link router-link-active">
                  Блог компании RUVDS.com
                </a></span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/hub/linux/" class="tm-article-body__tags-item-link">
                  Настройка Linux
                </a></span><span class="tm-article-body__tags-item"><a href="https://habr.com/ru/hub/sys_admin/" class="tm-article-body__tags-item-link">
                  Системное администрирование
                </a></span></div></div></article><div class="tm-article__icons-wrapper" style="display: none;"><div class="tm-data-icons tm-page-article__counters-panel"><div class="tm-article-rating tm-data-icons__item"><div class="tm-votes-meter tm-article-rating__votes-switcher"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_medium"><title>Всего голосов 26: ↑22 и ↓4</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 26: ↑22 и ↓4" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_medium">+18</span></div><div class="v-portal" style="display: none;"></div></div><!----><!----><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    290
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/company/ruvds/blog/328436/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      7
    </span></a><a href="https://habr.com/ru/company/ruvds/blog/328436/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +7
    </span></a></div><div class="tm-sharing tm-data-icons__item" title="Поделиться"><button type="button" class="tm-sharing__button"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" class="tm-sharing__icon"><path fill="currentColor" d="M10.33.275l9.047 7.572a.2.2 0 010 .306l-9.048 7.572a.2.2 0 01-.328-.153V11c-8 0-9.94 6-9.94 6S-1 5 10 5V.428a.2.2 0 01.328-.153z"></path></svg></button><!----></div><div class="v-portal" style="display: none;"></div></div></div><!----></div><!----><div class="tm-page-article__additional-blocks"><div class="tm-editoral-subscription"><span class="tm-svg-icon__wrapper tm-editoral-subscription__close"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Закрыть</title><use xlink:href="/img/megazord-v24.cee85629.svg#close"></use></svg></span><h3 class="tm-editoral-subscription__title">
    Редакторский дайджест
  </h3><p class="tm-editoral-subscription__description">
    Присылаем лучшие статьи раз&nbsp;в&nbsp;месяц
  </p><form method="GET" class="tm-editoral-subscription__form"><div class="tm-input-text-decorated tm-editoral-subscription__input tm-input-text-decorated_has-label-after" name="email" placeholder="Электропочта" type="email"><!----><input name="email" placeholder="Электропочта" type="email" class="tm-input-text-decorated__input"><div class="tm-input-text-decorated__label tm-input-text-decorated__label_after"><button type="submit" class="tm-editoral-subscription__icon"><svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" class=""><path fill="currentColor" clip-rule="evenodd" d="M19.707 11.293a1 1 0 010 1.414l-5.5 5.5a1 1 0 01-1.414-1.414L16.586 13H5a1 1 0 110-2h11.586l-3.793-3.793a1 1 0 011.414-1.414l5.5 5.5z"></path></svg></button></div></div><!----></form></div><section class="tm-block tm-block_spacing-bottom"><!----><div class="tm-block__body"><div class="tm-article-author tm-page-article__author"><div class="tm-article-author__company"><div class="tm-article-author__company-card"><div class="tm-company-snippet"><a href="https://habr.com/ru/company/ruvds/profile/" class="tm-company-snippet__logo-link"><div class="tm-entity-image"><img alt="" height="40" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/f27ea85bd62fbd985c308110885af7e0.png" width="40" class="tm-entity-image__pic"></div></a><div class="tm-company-snippet__info"><a href="https://habr.com/ru/company/ruvds/profile/" class="tm-company-snippet__title">RUVDS.com</a><div class="tm-company-snippet__description">VDS/VPS-хостинг. Скидка 10% по коду <b>HABR10</b></div></div></div><div class="tm-article-author__buttons"><!----><!----></div></div><div class="tm-article-author__company-contacts"><a href="https://facebook.com/ruvds" rel="noopener" target="_blank" class="tm-article-author__contact">
        Facebook
      </a><a href="https://twitter.com/ru_vds" rel="noopener" target="_blank" class="tm-article-author__contact">
        Twitter
      </a><a href="https://vk.com/ru_vds" rel="noopener" target="_blank" class="tm-article-author__contact">
        ВКонтакте
      </a></div><div class="tm-article-author__separator"></div></div><div class="tm-user-card tm-article-author__user-card tm-user-card_variant-two-column"><div class="tm-user-card__info-container"><div class="tm-user-card__header"><div class="tm-user-card__header-data"><a href="https://habr.com/ru/users/ru_vds/" class="tm-user-card__userpic tm-user-card__userpic_size-40"><div class="tm-entity-image"><img alt="" src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/e36812d3de38eb3d479d7a13228717ef(1).png" class="tm-entity-image__pic"></div></a><div class="tm-user-card__meta"><div title=" 943 голоса " class="tm-karma tm-user-card__karma"><div class="tm-karma__votes tm-karma__votes_positive">
    204.5
  </div><div class="tm-karma__text">
    Карма
  </div></div><div class="tm-rating tm-user-card__rating" title="Рейтинг пользователя"><div class="tm-rating__header"><div class="tm-rating__counter">110</div></div><div class="tm-rating__text">
    Рейтинг
  </div></div></div></div></div><div class="tm-user-card__info"><div class="tm-user-card__title"><!----><a href="https://habr.com/ru/users/ru_vds/" class="tm-user-card__nickname">
          @ru_vds
        </a><!----></div><p class="tm-user-card__short-info">Пользователь</p></div></div><div class="tm-user-card__buttons tm-user-card__buttons_variant-two-column"><!----><!----><!----><!----><!----></div></div><!----></div></div><!----></section><div class="tm-page-article__comments"><div class="tm-article-page-comments"><div class="tm-article-comments-counter-link tm-article-comments-counter-button"><a href="https://habr.com/ru/company/ruvds/blog/328436/comments/" class="tm-article-comments-counter-link__link tm-article-comments-counter-link__link_button-style"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon tm-article-comments-counter-link__icon_contrasted"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value tm-article-comments-counter-link__value_contrasted">
       Комментарии 7 
    </span></a><!----></div></div></div><!----><section class="tm-block tm-block_spacing-around"><header class="tm-block__header"><h2 class="tm-block__title">Похожие публикации</h2><!----></header><div class="tm-block__body"><ul class="tm-article-list-block__list"><li class="tm-article-list-block__item"><article class="tm-article-snippet-block-block tm-article-snippet-block-block_preview"><div class="tm-article-snippet-block__user-meta"><div class="tm-article-snippet-block__date"><time datetime="2020-12-09T09:47:23.000Z" title="2020-12-09, 16:47">9  декабря  2020 в 16:47</time></div></div><h2 class="tm-article-title tm-article-title_block"><a href="https://habr.com/ru/company/ruvds/blog/529840/" class="tm-article-title__link"><span>Кунг-фу стиля Linux: утилита marker и меню для командной строки</span></a></h2><div class="tm-data-icons"><!----><div class="tm-votes-meter tm-data-icons__item"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_small"><title>Всего голосов 29: ↑28 и ↓1</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 29: ↑28 и ↓1" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_small">+27</span></div><span class="tm-icon-counter tm-data-icons__item" title="Количество просмотров"><svg height="16" width="16" class="tm-svg-img tm-icon-counter__icon"><title>Просмотры</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-views"></use></svg><span class="tm-icon-counter__value">6.4K</span></span><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    64
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/company/ruvds/blog/529840/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      4
    </span></a><a href="https://habr.com/ru/company/ruvds/blog/529840/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +4
    </span></a></div><!----><div class="v-portal" style="display: none;"></div></div></article></li><li class="tm-article-list-block__item"><article class="tm-article-snippet-block-block tm-article-snippet-block-block_preview"><div class="tm-article-snippet-block__user-meta"><div class="tm-article-snippet-block__date"><time datetime="2017-10-11T09:07:52.000Z" title="2017-10-11, 16:07">11  октября  2017 в 16:07</time></div></div><h2 class="tm-article-title tm-article-title_block"><a href="https://habr.com/ru/company/ruvds/blog/339820/" class="tm-article-title__link"><span>20 приёмов работы в командной строке Linux, которые сэкономят уйму времени</span></a></h2><div class="tm-data-icons"><!----><div class="tm-votes-meter tm-data-icons__item"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_small"><title>Всего голосов 82: ↑46 и ↓36</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 82: ↑46 и ↓36" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_small">+10</span></div><span class="tm-icon-counter tm-data-icons__item" title="Количество просмотров"><svg height="16" width="16" class="tm-svg-img tm-icon-counter__icon"><title>Просмотры</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-views"></use></svg><span class="tm-icon-counter__value">167K</span></span><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    768
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/company/ruvds/blog/339820/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      87
    </span></a><a href="https://habr.com/ru/company/ruvds/blog/339820/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +87
    </span></a></div><!----><div class="v-portal" style="display: none;"></div></div></article></li><li class="tm-article-list-block__item"><article class="tm-article-snippet-block-block tm-article-snippet-block-block_preview"><div class="tm-article-snippet-block__user-meta"><div class="tm-article-snippet-block__date"><time datetime="2017-08-30T11:57:46.000Z" title="2017-08-30, 18:57">30  августа  2017 в 18:57</time></div></div><h2 class="tm-article-title tm-article-title_block"><a href="https://habr.com/ru/company/ruvds/blog/336764/" class="tm-article-title__link"><span>[в закладки] PDF-версия статей про Bash-скрипты</span></a></h2><div class="tm-data-icons"><!----><div class="tm-votes-meter tm-data-icons__item"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_small"><title>Всего голосов 41: ↑35 и ↓6</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 41: ↑35 и ↓6" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_small">+29</span></div><span class="tm-icon-counter tm-data-icons__item" title="Количество просмотров"><svg height="16" width="16" class="tm-svg-img tm-icon-counter__icon"><title>Просмотры</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-views"></use></svg><span class="tm-icon-counter__value">39K</span></span><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    510
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/company/ruvds/blog/336764/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      20
    </span></a><a href="https://habr.com/ru/company/ruvds/blog/336764/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +20
    </span></a></div><!----><div class="v-portal" style="display: none;"></div></div></article></li><!----></ul></div><!----></section><!----><section class="tm-block tm-block_spacing-around"><header class="tm-block__header"><h2 class="tm-block__title">Лучшие публикации за сутки</h2><!----></header><div class="tm-block__body"><ul class="tm-article-list-block__list"><li class="tm-article-list-block__item"><article class="tm-article-snippet-block-block tm-article-snippet-block-block_preview"><div class="tm-article-snippet-block__user-meta"><div class="tm-article-snippet-block__date"><time datetime="2021-08-26T09:00:02.000Z" title="2021-08-26, 16:00">вчера в 16:00</time></div></div><h2 class="tm-article-title tm-article-title_block"><a href="https://habr.com/ru/company/ruvds/blog/574742/" class="tm-article-title__link"><span>Сетевые соединения X11</span></a></h2><div class="tm-data-icons"><!----><div class="tm-votes-meter tm-data-icons__item"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_small"><title>Всего голосов 48: ↑46 и ↓2</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 48: ↑46 и ↓2" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_small">+44</span></div><span class="tm-icon-counter tm-data-icons__item" title="Количество просмотров"><svg height="16" width="16" class="tm-svg-img tm-icon-counter__icon"><title>Просмотры</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-views"></use></svg><span class="tm-icon-counter__value">5.3K</span></span><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    46
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/company/ruvds/blog/574742/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      9
    </span></a><a href="https://habr.com/ru/company/ruvds/blog/574742/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +9
    </span></a></div><!----><div class="v-portal" style="display: none;"></div></div></article></li><li class="tm-article-list-block__item"><article class="tm-article-snippet-block-block tm-article-snippet-block-block_preview"><div class="tm-article-snippet-block__user-meta"><div class="tm-article-snippet-block__date"><time datetime="2021-08-26T13:00:01.000Z" title="2021-08-26, 20:00">вчера в 20:00</time></div></div><h2 class="tm-article-title tm-article-title_block"><a href="https://habr.com/ru/company/ruvds/blog/574352/" class="tm-article-title__link"><span>Осваиваем LaTeX за 30 минут</span></a></h2><div class="tm-data-icons"><!----><div class="tm-votes-meter tm-data-icons__item"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_small"><title>Всего голосов 47: ↑43 и ↓4</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 47: ↑43 и ↓4" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_small">+39</span></div><span class="tm-icon-counter tm-data-icons__item" title="Количество просмотров"><svg height="16" width="16" class="tm-svg-img tm-icon-counter__icon"><title>Просмотры</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-views"></use></svg><span class="tm-icon-counter__value">5K</span></span><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    117
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/company/ruvds/blog/574352/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      27
    </span></a><a href="https://habr.com/ru/company/ruvds/blog/574352/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +27
    </span></a></div><!----><div class="v-portal" style="display: none;"></div></div></article></li><li class="tm-article-list-block__item"><article class="tm-article-snippet-block-block tm-article-snippet-block-block_preview"><div class="tm-article-snippet-block__user-meta"><div class="tm-article-snippet-block__date"><time datetime="2021-08-26T21:08:54.000Z" title="2021-08-27, 04:08">сегодня в 04:08</time></div></div><h2 class="tm-article-title tm-article-title_block"><a href="https://habr.com/ru/post/571514/" class="tm-article-title__link"><span>Самогонный аппарат — полный автомат. Часть 3. Заключительная</span></a></h2><div class="tm-data-icons"><!----><div class="tm-votes-meter tm-data-icons__item"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_small"><title>Всего голосов 38: ↑37 и ↓1</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 38: ↑37 и ↓1" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_small">+36</span></div><span class="tm-icon-counter tm-data-icons__item" title="Количество просмотров"><svg height="16" width="16" class="tm-svg-img tm-icon-counter__icon"><title>Просмотры</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-views"></use></svg><span class="tm-icon-counter__value">6.9K</span></span><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    50
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/post/571514/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      7
    </span></a><a href="https://habr.com/ru/post/571514/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +7
    </span></a></div><!----><div class="v-portal" style="display: none;"></div></div></article></li><li class="tm-article-list-block__item"><article class="tm-article-snippet-block-block tm-article-snippet-block-block_preview"><div class="tm-article-snippet-block__user-meta"><div class="tm-article-snippet-block__date"><time datetime="2021-08-26T17:24:20.000Z" title="2021-08-27, 00:24">сегодня в 00:24</time></div></div><h2 class="tm-article-title tm-article-title_block"><a href="https://habr.com/ru/company/kaspersky/blog/574884/" class="tm-article-title__link"><span>Sony Playstation Portable, радость коллекционера</span></a></h2><div class="tm-data-icons"><!----><div class="tm-votes-meter tm-data-icons__item"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_small"><title>Всего голосов 33: ↑31 и ↓2</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 33: ↑31 и ↓2" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_small">+29</span></div><span class="tm-icon-counter tm-data-icons__item" title="Количество просмотров"><svg height="16" width="16" class="tm-svg-img tm-icon-counter__icon"><title>Просмотры</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-views"></use></svg><span class="tm-icon-counter__value">6.5K</span></span><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    15
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/company/kaspersky/blog/574884/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      15
    </span></a><a href="https://habr.com/ru/company/kaspersky/blog/574884/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +15
    </span></a></div><!----><div class="v-portal" style="display: none;"></div></div></article></li><li class="tm-article-list-block__item"><article class="tm-article-snippet-block-block tm-article-snippet-block-block_preview"><div class="tm-article-snippet-block__user-meta"><div class="tm-article-snippet-block__date"><time datetime="2021-08-26T11:06:06.000Z" title="2021-08-26, 18:06">вчера в 18:06</time></div></div><h2 class="tm-article-title tm-article-title_block"><a href="https://habr.com/ru/company/ods/blog/572264/" class="tm-article-title__link"><span>Анализ вакансий и зарплат в Data Science</span></a></h2><div class="tm-data-icons"><!----><div class="tm-votes-meter tm-data-icons__item"><svg height="16" width="16" class="tm-svg-img tm-votes-meter__icon tm-votes-meter__icon_small"><title>Всего голосов 30: ↑29 и ↓1</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-rating"></use></svg><span title="Всего голосов 30: ↑29 и ↓1" class="tm-votes-meter__value tm-votes-meter__value_positive tm-votes-meter__value_small">+28</span></div><span class="tm-icon-counter tm-data-icons__item" title="Количество просмотров"><svg height="16" width="16" class="tm-svg-img tm-icon-counter__icon"><title>Просмотры</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-views"></use></svg><span class="tm-icon-counter__value">7.8K</span></span><button title="Добавить в закладки" type="button" class="bookmarks-button tm-data-icons__item"><span title="Добавить в закладки" class="tm-svg-icon__wrapper bookmarks-button__icon"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Добавить в закладки</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-favorite"></use></svg></span><span title="Количество пользователей, добавивших публикацию в закладки" class="bookmarks-button__counter">
    31
  </span></button><div class="tm-article-comments-counter-link tm-data-icons__item" title="Читать комментарии"><a href="https://habr.com/ru/company/ods/blog/572264/comments/" class="tm-article-comments-counter-link__link"><svg height="16" width="16" class="tm-svg-img tm-article-comments-counter-link__icon"><title>Комментарии</title><use xlink:href="/img/megazord-v24.cee85629.svg#counter-comments"></use></svg><span class="tm-article-comments-counter-link__value">
      11
    </span></a><a href="https://habr.com/ru/company/ods/blog/572264/comments/" class="tm-article-comments-counter-link__link"><span title="Читать новые комментарии" class="tm-article-comments-counter-link__unread-counter">
      +11
    </span></a></div><!----><div class="v-portal" style="display: none;"></div></div></article></li><!----></ul></div><!----></section><!----><!----></div></div></div><div class="tm-page__sidebar"><!----></div></div></div></div></main><!----></div><!----><div class="tm-footer"><div class="tm-page-width"><div class="tm-footer__container"><div class="tm-footer__title"><a href="https://habr.com/ru/" class="tm-svg-icon__wrapper tm-footer__title-link router-link-active"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Хабр</title><use xlink:href="/img/habr-logo-ru.svg#logo"></use></svg></a></div><div class="tm-footer__social"><a class="tm-svg-icon__wrapper tm-social-icons__icon" href="https://www.facebook.com/habrahabr.ru" rel="nofollow noopener noreferrer" target="_blank"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Facebook</title><use xlink:href="/img/social-icons-sprite.svg#social-logo-facebook"></use></svg></a><a class="tm-svg-icon__wrapper tm-social-icons__icon" href="https://twitter.com/habr_com" rel="nofollow noopener noreferrer" target="_blank"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Twitter</title><use xlink:href="/img/social-icons-sprite.svg#social-logo-twitter"></use></svg></a><a class="tm-svg-icon__wrapper tm-social-icons__icon" href="https://vk.com/habr" rel="nofollow noopener noreferrer" target="_blank"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>VK</title><use xlink:href="/img/social-icons-sprite.svg#social-logo-vkontakte"></use></svg></a><a class="tm-svg-icon__wrapper tm-social-icons__icon" href="https://telegram.me/habr_com" rel="nofollow noopener noreferrer" target="_blank"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Telegram</title><use xlink:href="/img/social-icons-sprite.svg#social-logo-telegram"></use></svg></a><a class="tm-svg-icon__wrapper tm-social-icons__icon" href="https://www.youtube.com/channel/UCd_sTwKqVrweTt4oAKY5y4w" rel="nofollow noopener noreferrer" target="_blank"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Youtube</title><use xlink:href="/img/social-icons-sprite.svg#social-logo-youtube"></use></svg></a><a class="tm-svg-icon__wrapper tm-social-icons__icon" href="https://zen.yandex.ru/habr" rel="nofollow noopener noreferrer" target="_blank"><svg height="16" width="16" class="tm-svg-img tm-svg-icon"><title>Яндекс Дзен</title><use xlink:href="/img/social-icons-sprite.svg#social-logo-zen"></use></svg></a></div><div class="v-portal" style="display: none;"></div><button class="tm-footer__link"><svg height="16" width="16" class="tm-svg-img tm-footer__icon"><title>Язык</title><use xlink:href="/img/megazord-v24.cee85629.svg#lang"></use></svg>
        Настройка языка
      </button><a href="https://habr.com/ru/about" class="tm-footer__link">
        О сайте
      </a><a href="https://habr.com/ru/feedback/" class="tm-footer__link">
        Техническая поддержка
      </a><a href="https://habr.com/ru/company/ruvds/blog/328436/?mobile=no" class="tm-footer__link">
        Полная версия
      </a><a href="https://habr.com/berserk-mode-nope" class="tm-footer__link">
        Вернуться на старую версию
      </a><div class="tm-footer-copyright"><span class="tm-copyright"><span class="tm-copyright__years">© 2006–2021 </span><span class="tm-copyright__name">«<a href="https://company.habr.com/" rel="noopener" target="_blank" class="tm-copyright__link">Habr</a>»</span></span></div></div></div></div><!----><!----></div><div class="vue-portal-target"><!----></div></div>

<script src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/chunk-vendors.1ca41abb.js" defer=""></script><script src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/about.2d3fd602.js" defer=""></script><script src="./Bash-скрипты, часть 11_ expect и автоматизация интерактивных утилит _ Блог компании RUVDS.com _ Хабр_files/app.132461d8.js" defer=""></script>



    <script>
      (function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
        (i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),
        m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
      })(window,document,'script','//www.google-analytics.com/analytics.js','ga');
    </script>
  
  <script type="text/javascript">
    (function(m,e,t,r,i,k,a){m[i]=m[i]||function(){(m[i].a=m[i].a||[]).push(arguments)};
    m[i].l=1*new Date();k=e.createElement(t),a=e.getElementsByTagName(t)[0],k.async=1,k.src=r,a.parentNode.insertBefore(k,a)})
    (window, document, "script", "https://mc.yandex.ru/metrika/tag.js", "ym");

    ym(24049213, "init", {
      defer:true,
      trackLinks:true,
      accurateTrackBounce:true,
      webvisor:false,
    });
  </script>
  <noscript>
    <div>
      <img src="https://mc.yandex.ru/watch/24049213" style="position:absolute; left:-9999px;" alt="" />
    </div>
  </noscript>
  
    <script type="text/javascript">
      window.addEventListener('load', function () {
        setTimeout(() => {
          const img = new Image();
          img.src = 'https://vk.com/rtrg?p=VK-RTRG-421343-57vKE';
        }, 0);
      });
    </script>
  


<div><div><div class="Vue-Toastification__container top-left"></div></div><div><div class="Vue-Toastification__container top-center"></div></div><div><div class="Vue-Toastification__container top-right"></div></div><div><div class="Vue-Toastification__container bottom-left"></div></div><div><div class="Vue-Toastification__container bottom-center"></div></div><div><div class="Vue-Toastification__container bottom-right"></div></div></div></body></html>